package com.webserver.http;

import java.io.IOException;
import java.io.InputStream;
import java.io.UnsupportedEncodingException;
import java.net.Socket;
import java.net.URLDecoder;
import java.util.HashMap;
import java.util.Map;

/**
 * 请求对象
 * 该类的每一个实例用于表示客户端发送给浏览器的一个HTTP请求
 * HTTP协议规定一个请求包含三部分内容
 * 请求行,消息头,消息正文
 */
public class HttpRequest {
    //请求行相关信息
    private String method;//请求方式
    private String uri;//抽象路径
    private String protocol;//协议版本
    private String requestURI;//抽象路径中的请求部分
    private String queryString;//抽象路径中的参数部分
    private Map<String, String> parameters = new HashMap<>();

    //消息头相关信息
    //存放所有消息头的Map,key:消息头的名字 value:消息头的值
    private Map<String, String> headers = new HashMap<>();

    //消息正文相关信息

    private Socket socket;

    /**
     * 实例化HttpRequest的过程就是解析请求的过程
     */
    public HttpRequest(Socket socket) throws IOException, EmptyRequestExpetion {
        this.socket = socket;
        //1解析请求行
        parseRequestLine();
        //2解析消息头
        parseHeaders();
        //3解析消息正文
        parseContent();
    }

    /**
     * 解析请求行
     */
    public void parseRequestLine() throws IOException, EmptyRequestExpetion {
        String line = readLine();
        if (line.isEmpty()){//请求行如果没有内容则说明是一个空请求
            throw  new EmptyRequestExpetion();
        }
        System.out.println("请求行:" + line);

        String[] sp = line.split("\\s");
        method = sp[0];
        uri = sp[1];
        protocol = sp[2];
        parseUri();//进一步解析uri
        System.out.println("method:" + method);
        System.out.println("uri:" + uri);
        System.out.println("protocol:" + protocol);
    }

    /**
     * 进一步解析uri
     */
    private void parseUri() {
        /*
            uri分为两种情况:含有参数和不含有参数
            如果uri不含有参数,则直接将uri的值赋值给requestURI即可

            如果uri中含有参数,则需要进一步拆分解析:
            首先按照"?"将uri拆分为两部分:请求部分和参数部分
            然后将请求部分赋值给requestURI
            参数部分赋值给queryString
            之后再进一步拆分出每一组参数
            将参数部分按照"&"进行拆分,每一组参数再按照"="拆分出参数名
            与参数值.并将参数名作为key,参数值作为value存入parameters
            这个Map中完成解析
         */
        if (!uri.contains("?")) {
            requestURI = uri;
        } else {
            String[] line = uri.split("\\?");
            requestURI = line[0];
            if (line.length > 1) {
                queryString = line[1];
                pareseParameter(queryString);
            }
        }
        System.out.println("requestURI:" + requestURI);
        System.out.println("queryString:" + queryString);
        System.out.println("parameters:" + parameters);
    }

    /**
     * 解析参数
     * @param line
     */
    private void pareseParameter(String line){
        String[] line1 = line.split("&");
        for (String str : line1) {
            String[] str1 = str.split("=");
            if (str1.length > 1) {
                try {
                    str1[1] = URLDecoder.decode(str1[1],"UTF-8");
                } catch (UnsupportedEncodingException e) {
                    e.printStackTrace();
                }
                parameters.put(str1[0], str1[1]);
            } else {
                parameters.put(str1[0], null);
            }
        }
    }

    /**
     * 解析消息头
     */
    public void parseHeaders() throws IOException {
        while (true) {
            String line = readLine();
            //读取请求行时如果返回的是空字符串,说明单独读取到了CRLF
            if (line.isEmpty()) {
                break;
            }
            String[] str = line.split("(:\\s)");
            headers.put(str[0], str[1]);

        }
        headers.forEach((s, s2) -> System.out.println("消息名:" + s + ",消息值:" + s2));
//            System.out.println("headers"+headers);
    }

    /**
     * 解析消息正文
     */
    public void parseContent() throws IOException {
        //判断请求方式是否为post
        if ("post".equalsIgnoreCase(method)){
            //通过消息头Content-Length取得消息正文的长度(确定字节量)
            if (headers.containsKey("Content-Length")){
                int len = Integer.parseInt(
                        headers.get("Content-Length"));
                //根据正文长度将正文内容读取到数组中备用
                byte[] data = new byte[len];
                InputStream in = socket.getInputStream();
                in.read(data);
                //根据消息头Content-Type来处理消息正文
                String  contentType = headers.get("Content-Type");
                if ("application/x-www-form-urlencoded".equals(contentType)){
                    //判定该正文就是一个字符串,原GET请求提交是URL中"?"右侧内容
                    String line = new String(data,"ISO8859-1");
                    pareseParameter(line);
                    System.out.println("正文内容:"+line);
                }
            }
        }
    }

    public String readLine() throws IOException {
        /*
            调用同一个socket的getInputStream方法时,无论调用多少次
            返回的输入流始终是同一个流,输出流也是一样的.
         */
        InputStream in = socket.getInputStream();
        StringBuilder str = new StringBuilder();
        //cur表示本次读取到的字符,pre表示上一次读取到的字符
        char cur = 'a', pre = 'a';
        int d;
        while ((d = in.read()) != -1) {
            cur = (char) d;
            if (pre == 13 && cur == 10) {
                break;
            }
            str.append(cur);
            pre = cur;
        }
        return str.toString().trim();
    }

    public String getMethod() {
        return method;
    }

    public String getUri() {
        return uri;
    }

    public String getProtocol() {
        return protocol;
    }

    public String getHeader(String name) {
        return headers.get(name);
    }

    public String getRequestURI() {
        return requestURI;
    }

    public String getQueryString() {
        return queryString;
    }

    public String getParameter(String name) {
        return parameters.get(name);
    }
}
